| Conditions | 4 |
| Paths | 2 |
| Total Lines | 142 |
| Code Lines | 83 |
| Lines | 1 |
| Ratio | 0.7 % |
| Changes | 0 | ||
Small methods make your code easier to understand, in particular if combined with a good name. Besides, if your method is small, finding a good name is usually much easier.
For example, if you find yourself adding comments to a method's body, this is usually a good sign to extract the commented part to a new method, and use the comment as a starting point when coming up with a good name for this new method.
Commonly applied refactorings include:
If many parameters/temporary variables are present:
| 1 | /*istanbul ignore start*/'use strict'; |
||
| 5 | View Code Duplication | function parsePatch(uniDiff) { |
|
|
|
|||
| 6 | /*istanbul ignore start*/var /*istanbul ignore end*/options = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {}; |
||
| 7 | |||
| 8 | var diffstr = uniDiff.split(/\r\n|[\n\v\f\r\x85]/), |
||
| 9 | delimiters = uniDiff.match(/\r\n|[\n\v\f\r\x85]/g) || [], |
||
| 10 | list = [], |
||
| 11 | i = 0; |
||
| 12 | |||
| 13 | function parseIndex() { |
||
| 14 | var index = {}; |
||
| 15 | list.push(index); |
||
| 16 | |||
| 17 | // Parse diff metadata |
||
| 18 | while (i < diffstr.length) { |
||
| 19 | var line = diffstr[i]; |
||
| 20 | |||
| 21 | // File header found, end parsing diff metadata |
||
| 22 | if (/^(\-\-\-|\+\+\+|@@)\s/.test(line)) { |
||
| 23 | break; |
||
| 24 | } |
||
| 25 | |||
| 26 | // Diff index |
||
| 27 | var header = /^(?:Index:|diff(?: -r \w+)+)\s+(.+?)\s*$/.exec(line); |
||
| 28 | if (header) { |
||
| 29 | index.index = header[1]; |
||
| 30 | } |
||
| 31 | |||
| 32 | i++; |
||
| 33 | } |
||
| 34 | |||
| 35 | // Parse file headers if they are defined. Unified diff requires them, but |
||
| 36 | // there's no technical issues to have an isolated hunk without file header |
||
| 37 | parseFileHeader(index); |
||
| 38 | parseFileHeader(index); |
||
| 39 | |||
| 40 | // Parse hunks |
||
| 41 | index.hunks = []; |
||
| 42 | |||
| 43 | while (i < diffstr.length) { |
||
| 44 | var _line = diffstr[i]; |
||
| 45 | |||
| 46 | if (/^(Index:|diff|\-\-\-|\+\+\+)\s/.test(_line)) { |
||
| 47 | break; |
||
| 48 | } else if (/^@@/.test(_line)) { |
||
| 49 | index.hunks.push(parseHunk()); |
||
| 50 | } else if (_line && options.strict) { |
||
| 51 | // Ignore unexpected content unless in strict mode |
||
| 52 | throw new Error('Unknown line ' + (i + 1) + ' ' + JSON.stringify(_line)); |
||
| 53 | } else { |
||
| 54 | i++; |
||
| 55 | } |
||
| 56 | } |
||
| 57 | } |
||
| 58 | |||
| 59 | // Parses the --- and +++ headers, if none are found, no lines |
||
| 60 | // are consumed. |
||
| 61 | function parseFileHeader(index) { |
||
| 62 | var headerPattern = /^(---|\+\+\+)\s+([\S ]*)(?:\t(.*?)\s*)?$/; |
||
| 63 | var fileHeader = headerPattern.exec(diffstr[i]); |
||
| 64 | if (fileHeader) { |
||
| 65 | var keyPrefix = fileHeader[1] === '---' ? 'old' : 'new'; |
||
| 66 | var fileName = fileHeader[2].replace(/\\\\/g, '\\'); |
||
| 67 | if (/^".*"$/.test(fileName)) { |
||
| 68 | fileName = fileName.substr(1, fileName.length - 2); |
||
| 69 | } |
||
| 70 | index[keyPrefix + 'FileName'] = fileName; |
||
| 71 | index[keyPrefix + 'Header'] = fileHeader[3]; |
||
| 72 | |||
| 73 | i++; |
||
| 74 | } |
||
| 75 | } |
||
| 76 | |||
| 77 | // Parses a hunk |
||
| 78 | // This assumes that we are at the start of a hunk. |
||
| 79 | function parseHunk() { |
||
| 80 | var chunkHeaderIndex = i, |
||
| 81 | chunkHeaderLine = diffstr[i++], |
||
| 82 | chunkHeader = chunkHeaderLine.split(/@@ -(\d+)(?:,(\d+))? \+(\d+)(?:,(\d+))? @@/); |
||
| 83 | |||
| 84 | var hunk = { |
||
| 85 | oldStart: +chunkHeader[1], |
||
| 86 | oldLines: +chunkHeader[2] || 1, |
||
| 87 | newStart: +chunkHeader[3], |
||
| 88 | newLines: +chunkHeader[4] || 1, |
||
| 89 | lines: [], |
||
| 90 | linedelimiters: [] |
||
| 91 | }; |
||
| 92 | |||
| 93 | var addCount = 0, |
||
| 94 | removeCount = 0; |
||
| 95 | for (; i < diffstr.length; i++) { |
||
| 96 | // Lines starting with '---' could be mistaken for the "remove line" operation |
||
| 97 | // But they could be the header for the next file. Therefore prune such cases out. |
||
| 98 | if (diffstr[i].indexOf('--- ') === 0 && i + 2 < diffstr.length && diffstr[i + 1].indexOf('+++ ') === 0 && diffstr[i + 2].indexOf('@@') === 0) { |
||
| 99 | break; |
||
| 100 | } |
||
| 101 | var operation = diffstr[i][0]; |
||
| 102 | |||
| 103 | if (operation === '+' || operation === '-' || operation === ' ' || operation === '\\') { |
||
| 104 | hunk.lines.push(diffstr[i]); |
||
| 105 | hunk.linedelimiters.push(delimiters[i] || '\n'); |
||
| 106 | |||
| 107 | if (operation === '+') { |
||
| 108 | addCount++; |
||
| 109 | } else if (operation === '-') { |
||
| 110 | removeCount++; |
||
| 111 | } else if (operation === ' ') { |
||
| 112 | addCount++; |
||
| 113 | removeCount++; |
||
| 114 | } |
||
| 115 | } else { |
||
| 116 | break; |
||
| 117 | } |
||
| 118 | } |
||
| 119 | |||
| 120 | // Handle the empty block count case |
||
| 121 | if (!addCount && hunk.newLines === 1) { |
||
| 122 | hunk.newLines = 0; |
||
| 123 | } |
||
| 124 | if (!removeCount && hunk.oldLines === 1) { |
||
| 125 | hunk.oldLines = 0; |
||
| 126 | } |
||
| 127 | |||
| 128 | // Perform optional sanity checking |
||
| 129 | if (options.strict) { |
||
| 130 | if (addCount !== hunk.newLines) { |
||
| 131 | throw new Error('Added line count did not match for hunk at line ' + (chunkHeaderIndex + 1)); |
||
| 132 | } |
||
| 133 | if (removeCount !== hunk.oldLines) { |
||
| 134 | throw new Error('Removed line count did not match for hunk at line ' + (chunkHeaderIndex + 1)); |
||
| 135 | } |
||
| 136 | } |
||
| 137 | |||
| 138 | return hunk; |
||
| 139 | } |
||
| 140 | |||
| 141 | while (i < diffstr.length) { |
||
| 142 | parseIndex(); |
||
| 143 | } |
||
| 144 | |||
| 145 | return list; |
||
| 146 | } |
||
| 147 | //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/patch/parse.js"],"names":["parsePatch","uniDiff","options","diffstr","split","delimiters","match","list","i","parseIndex","index","push","length","line","test","header","exec","parseFileHeader","hunks","parseHunk","strict","Error","JSON","stringify","headerPattern","fileHeader","keyPrefix","fileName","replace","substr","chunkHeaderIndex","chunkHeaderLine","chunkHeader","hunk","oldStart","oldLines","newStart","newLines","lines","linedelimiters","addCount","removeCount","indexOf","operation"],"mappings":"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","file":"parse.js","sourcesContent":["export function parsePatch(uniDiff, options = {}) {\n  let diffstr = uniDiff.split(/\\r\\n|[\\n\\v\\f\\r\\x85]/),\n      delimiters = uniDiff.match(/\\r\\n|[\\n\\v\\f\\r\\x85]/g) || [],\n      list = [],\n      i = 0;\n\n  function parseIndex() {\n    let index = {};\n    list.push(index);\n\n    // Parse diff metadata\n    while (i < diffstr.length) {\n      let line = diffstr[i];\n\n      // File header found, end parsing diff metadata\n      if (/^(\\-\\-\\-|\\+\\+\\+|@@)\\s/.test(line)) {\n        break;\n      }\n\n      // Diff index\n      let header = (/^(?:Index:|diff(?: -r \\w+)+)\\s+(.+?)\\s*$/).exec(line);\n      if (header) {\n        index.index = header[1];\n      }\n\n      i++;\n    }\n\n    // Parse file headers if they are defined. Unified diff requires them, but\n    // there's no technical issues to have an isolated hunk without file header\n    parseFileHeader(index);\n    parseFileHeader(index);\n\n    // Parse hunks\n    index.hunks = [];\n\n    while (i < diffstr.length) {\n      let line = diffstr[i];\n\n      if (/^(Index:|diff|\\-\\-\\-|\\+\\+\\+)\\s/.test(line)) {\n        break;\n      } else if (/^@@/.test(line)) {\n        index.hunks.push(parseHunk());\n      } else if (line && options.strict) {\n        // Ignore unexpected content unless in strict mode\n        throw new Error('Unknown line ' + (i + 1) + ' ' + JSON.stringify(line));\n      } else {\n        i++;\n      }\n    }\n  }\n\n  // Parses the --- and +++ headers, if none are found, no lines\n  // are consumed.\n  function parseFileHeader(index) {\n    const headerPattern = /^(---|\\+\\+\\+)\\s+([\\S ]*)(?:\\t(.*?)\\s*)?$/;\n    const fileHeader = headerPattern.exec(diffstr[i]);\n    if (fileHeader) {\n      let keyPrefix = fileHeader[1] === '---' ? 'old' : 'new';\n      let fileName = fileHeader[2].replace(/\\\\\\\\/g, '\\\\');\n      if (/^\".*\"$/.test(fileName)) {\n        fileName = fileName.substr(1, fileName.length - 2);\n      }\n      index[keyPrefix + 'FileName'] = fileName;\n      index[keyPrefix + 'Header'] = fileHeader[3];\n\n      i++;\n    }\n  }\n\n  // Parses a hunk\n  // This assumes that we are at the start of a hunk.\n  function parseHunk() {\n    let chunkHeaderIndex = i,\n        chunkHeaderLine = diffstr[i++],\n        chunkHeader = chunkHeaderLine.split(/@@ -(\\d+)(?:,(\\d+))? \\+(\\d+)(?:,(\\d+))? @@/);\n\n    let hunk = {\n      oldStart: +chunkHeader[1],\n      oldLines: +chunkHeader[2] || 1,\n      newStart: +chunkHeader[3],\n      newLines: +chunkHeader[4] || 1,\n      lines: [],\n      linedelimiters: []\n    };\n\n    let addCount = 0,\n        removeCount = 0;\n    for (; i < diffstr.length; i++) {\n      // Lines starting with '---' could be mistaken for the \"remove line\" operation\n      // But they could be the header for the next file. Therefore prune such cases out.\n      if (diffstr[i].indexOf('--- ') === 0\n            && (i + 2 < diffstr.length)\n            && diffstr[i + 1].indexOf('+++ ') === 0\n            && diffstr[i + 2].indexOf('@@') === 0) {\n          break;\n      }\n      let operation = diffstr[i][0];\n\n      if (operation === '+' || operation === '-' || operation === ' ' || operation === '\\\\') {\n        hunk.lines.push(diffstr[i]);\n        hunk.linedelimiters.push(delimiters[i] || '\\n');\n\n        if (operation === '+') {\n          addCount++;\n        } else if (operation === '-') {\n          removeCount++;\n        } else if (operation === ' ') {\n          addCount++;\n          removeCount++;\n        }\n      } else {\n        break;\n      }\n    }\n\n    // Handle the empty block count case\n    if (!addCount && hunk.newLines === 1) {\n      hunk.newLines = 0;\n    }\n    if (!removeCount && hunk.oldLines === 1) {\n      hunk.oldLines = 0;\n    }\n\n    // Perform optional sanity checking\n    if (options.strict) {\n      if (addCount !== hunk.newLines) {\n        throw new Error('Added line count did not match for hunk at line ' + (chunkHeaderIndex + 1));\n      }\n      if (removeCount !== hunk.oldLines) {\n        throw new Error('Removed line count did not match for hunk at line ' + (chunkHeaderIndex + 1));\n      }\n    }\n\n    return hunk;\n  }\n\n  while (i < diffstr.length) {\n    parseIndex();\n  }\n\n  return list;\n}\n"]} |
||
| 148 |